MySQL中查询一个字符串字段的值不为空到底该怎么写? 您所在的位置:网站首页 sql 数据不为空 MySQL中查询一个字符串字段的值不为空到底该怎么写?

MySQL中查询一个字符串字段的值不为空到底该怎么写?

2024-07-01 08:46| 来源: 网络整理| 查看: 265

如果你看过SQL开发规范,肯定知道尽量 不要对字段使用函数

但是,就好像三大范式中我只遵循第一范式而忽略后面两个范式一样,千万不要教条!

如果你要查询表中指定的字符串类型的字段的值不为空时,通常都是 column_name != '' ,也可能有人会写 LENGTH(column_name) > 0 ,或者 CHAR_LENGTH(column_name) > 0

先看一下构造数据的SQL:

DROP TABLE IF EXISTS `not_empty_query`; CREATE TABLE `not_empty_query` ( `id` INT UNSIGNED NOT NULL AUTO_INCREMENT, `related_id` varchar(50) NOT NULL DEFAULT '', PRIMARY KEY(`id`) ) ENGINE=InnoDB COMMENT '查询字段值不为空'; -- 创建填充数据的存储过程 DROP PROCEDURE IF EXISTS `mysp_fill_table_not_empty_query`; DELIMITER $$ CREATE PROCEDURE `mysp_fill_table_not_empty_query`(in n int) BEGIN DECLARE i INT UNSIGNED; SET i = 0; START TRANSACTION; WHILE i < n DO INSERT INTO `not_empty_query`(`related_id`)VALUES(uuid()); SET i = i + 1; END WHILE; COMMIT; END $$ DELIMITER ; -- 调用存储过程, 插入100w行数据 CALL `mysp_fill_table_not_empty_query`(1000000); DROP PROCEDURE IF EXISTS `mysp_fill_table_not_empty_query`;

上面的代码向表 not_empty_query 中插入了100万行数据。(代码中产生的每行的长度都是一样,这里只是为了方便)

然后来看下3种查询字段值不为空的SQL的explain结果:

mysql> EXPLAIN SELECT SQL_NO_CACHE COUNT(*) FROM `not_empty_query` WHERE `related_id` != ''; +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | 1 | SIMPLE | not_empty_query | NULL | ALL | NULL | NULL | NULL | NULL | 996442 | 90.00 | Using where | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ 1 row in set, 2 warnings (0.00 sec) mysql> EXPLAIN SELECT SQL_NO_CACHE COUNT(*) FROM `not_empty_query` WHERE LENGTH(`related_id`) > 0; +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | 1 | SIMPLE | not_empty_query | NULL | ALL | NULL | NULL | NULL | NULL | 996442 | 100.00 | Using where | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ 1 row in set, 2 warnings (0.00 sec) mysql> EXPLAIN SELECT SQL_NO_CACHE COUNT(*) FROM `not_empty_query` WHERE CHAR_LENGTH(`related_id`) > 0; +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ | 1 | SIMPLE | not_empty_query | NULL | ALL | NULL | NULL | NULL | NULL | 996442 | 100.00 | Using where | +----+-------------+-----------------+------------+------+---------------+------+---------+------+--------+----------+-------------+ 1 row in set, 2 warnings (0.00 sec)

唯一的差别在于 filtered 字段, 只有第一个 related_id != '' 是 90.00。这个其实是没有多大意义的。(更多信息可参考 官方文档 及 MySQL explain中的filtered的意义 )

根据MySQL存储数据的格式我们知道,每一行数据的所有字段的内容是连续存储在一起的的(text等类型除外),对于int、char(n)这种定长字段,我们从字段定义中就知道它们的长度,对于varchar(n)这种只有最大长度的字符串字段,必须存储其字节长度值。

对于这三种判断字段是否为空的方法,其处理过程应该是:

column_name != '' 取出字段的值, 然后和空字符串进行比较 LENGTH(column_name) > 0 从行数据的元数据中直接取出字节长度值, 然后与0比较 CHAR_LENGTH(column_name) > 0 取出字段的值, 然后根据CHARSET计算字符串长度, 然后与0比较

根据上面的分析, 理论认为其查询速度是: 2 > 1 > 3

我们在本地多次执行的结果, 与理论分析一致:

方式 4次执行时间(s) 1. column_name != '' 0.19, 0.19, 0.19, 0.19 2. LENGTH(column_name) > 0 0.17, 0.17, 0.16, 0.16 3. CHAR_LENGTH(column_name) > 0 0.25, 0.25, 0.25, 0.25

当然, 如果表的字段比较多,或者总长度比较大,或者包含了text等类型的字段,则情况又更为复杂,这里暂不讨论。



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有